package thread.unit2.learntsafe;


/**
 * 线程安全问题
 * 如何解决 卖票系统的线程安全问题
 *
 * 1.问题：卖票过程中，出现了重票、错票 -->出现了线程的安全问题
 * 2.问题出现的原因：当某个线程操作车票的过程中，尚未操作完成时，其他线程参与进来，也操作车票。
 * 3.如何解决：当一个线程a在操作ticket的时候，其他线程不能参与进来。直到线程a操作完ticket时，其他
 * 		 线程才可以开始操作ticket。这种情况即使线程a出现了阻塞，也不能被改变。
 *
 *
 * 4.在Java中，我们通过同步机制，来解决线程的安全问题。
 *
 * 方式一：同步代码块
 *
 * synchronized(同步监视器){
 *    //需要被同步的代码
 *
 * }
 * 说明：1.操作共享数据的代码，即为需要被同步的代码。  -->不能包含代码多了，也不能包含代码少了。根据业务场景判断
 * 	2.共享数据：多个线程共同操作的变量。比如：本题中的ticket就是共享数据。
 * 	3.同步监视器，俗称：锁。任何一个类的对象，都可以充当锁。
 * 	   要求：多个线程必须要共用同一把锁。
 *
 * 	补充：在实现Runnable接口创建多线程的方式中，我们可以考虑使用this充当同步监视器。
 * 方式二：同步方法。
 *   如果操作共享数据的代码完整的声明在一个方法中，我们不妨将此方法声明同步的。
 *
 *
 * 5.同步的方式，解决了线程的安全问题。---好处
 *  操作同步代码时，只能有一个线程参与，其他线程等待。相当于是一个单线程的过程，效率低。 ---局限性
 */

/**
 *
 * 使用同步代码块方式 解决接口多线程的线程安全问题
 *
 */
public class ThreadSafeDemo1 implements Runnable {
    private int ticket = 51;
    private Object object = new Object();

    public static void main(String[] args) {
        ThreadSafeDemo1 threadP3 = new ThreadSafeDemo1();
        Thread w1 = new Thread(threadP3,"窗口一");
        Thread w2 = new Thread(threadP3,"窗口二");
        Thread w3 = new Thread(threadP3,"窗口三");
        w1.start();
        w2.start();
        w3.start();
    }

    @Override
    public void run() {
        while (true) {
            //此时的this会动态获取对象，在本题中this代表对象threadP3是唯一的，可以满足一把锁的需求
            synchronized(this){
//            synchronized (object) {
                if (ticket > 0) {
                    System.out.println(Thread.currentThread().getName() + "卖票，当前票号：" + ticket);
                    try {
                        Thread.sleep(100);
                    } catch (InterruptedException e) {
                        e.printStackTrace();
                    }
                    ticket--;

                } else {
                    break;
                }
            }
        }
    }
}
